

PL/M-86 COMPILER

VICTOR Business Products, Inc. (c) 1982 V9000 Hardware  
Example software drivers for V9000 Hardware

04/01/82

PAGE 1

SERIES-III PL/M-86 V1.0 COMPILED MODULE HARDWARE

NO OBJECT MODULE REQUESTED

COMPILER INVOKED BY: P.86 TEMP.SRC OPTIMIZE(3) PAGELENGTH(42) PAGEWIDTH(109) PRINT(:F4:HW.LS) NOOBJECT

```
$TITLE ('SIRIUS Victor Business Products (c) 1982 V9000 Hardware')
$SUBTITLE ('Example software drivers for V9000 Hardware')
*****
* Proprietary Rights Notice: All rights reserved. This material *
* contains the valuable properties and trade secrets of *
* Victor Business Products, Inc. (VICTOR) *
* of Chicago, Illinois, USA. *
* embodying substantial creative efforts and confidential *
* information, ideas, and expressions; no part of which may be *
* reproduced or transmitted in any form or by any means; electronic, *
* mechanical, or otherwise; including photo copying and recording *
* or in connection with any information storage or retrieval *
* system without the permission in writing from SIRIUS. *
* Copyright notice: Copyright (c) 1982 *
* An unpublished work by: *
* Victor Business Products, Inc. *
* 3900 N. Rockwell Street *
* Chicago, IL 60618 *
******/
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$eject
$SMALL ROM

1      . Hardware: do;

2 1      Declare dcl      literally 'declare';
3 1      Dcl     lit      literally 'literally';
4 1      Dcl     addr     lit      'address',
                  ext     lit      'external',
                  init    lit      'initial',
                  intg    lit      'integer',
                  proc    lit      'procedure',
                  ptr     lit      'pointer',
                  pub     lit      'public',
                  rent    lit      'reentrant',
                  ret     lit      'return',
                  struc   lit      'structure',
                  boolean lit      'byte',
                  true    lit      'OFFH',
                  false   lit      '0000H';
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$subtitle("KB: Hardware bit defs")
```

```
5 1 dcl SR$intbit      lit '4'; /* KB shift register interrupt mask in 6522 IER/IFR */
6 1 dcl SR$enable       lit '0ch'; /* KB shift register enable in 6522 ACR */
7 1 dcl CB1$intbit     lit '10h'; /* KB RDY edge-sense interrupt mask 6522 PCR */
8 1 dcl CB1$pos_edge   lit '10h'; /* KB RDY edge-sense control in 6522 PCR */

9 1 dcl kb$databit     lit '40h'; /* KB DATA level */
10 1 dcl kb$ackctl      lit '2'; /* KB ACK control for 6522 output */
11 1 dcl kb$TIMEOUT    lit '300'; /* error timeout in milliseconds */
12 1 dcl timer1_ena     lit '0c0h'; /* timer 1 interrupt mask in 6522 IER/IFR */
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

**\$eject**

/\* KYBRD PORT (e8040..e804f) \*/

```
13 1     dcl via(16) struc(                                /* 6522 port organization */  
          RB      byte,  
          RA      byte,  
          DDRB    byte,  
          DDRA    byte,  
          TIMER1  word,  
          TIMER1L word,  
          TIMER2  word,  
          SR      byte,  
          ACR     byte,  
          PCR     byte,  
          IFR     byte,  
          IER     byte,  
          RAX     byte) at(0e8000h);  
  
14 1     dcl kb$state           byte;                  /* current state of keyboard stateware */  
15 1     dcl kb$data            byte;                  /* constructed data from keyboard */  
  
16 1     dcl Ctable(*) byte data (0,8,4,0ch, 2,0ah,6,0eh, 1,9,5,0dh, 3,0bh,7,0fh);  
17 1     dcl tick               lit '50';             /* console clock rate in milliseconds */
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$subtitle("KB: external routines")

/*
 *   signal user about keyboard error state -- ring bell
 */
18 1     dcl signal$KB$error lit 'Ringbell';
           /* Ringbell found in SOUND module */

/*
 *   Process key board event -- in external module
 */
19 1     Process$Event: proc(event) byte ext;
20 2         dcl event byte;
21 2         end;

/*
 *   Software clock resource -- set timeout for interrupt to KB$reset
 */
22 1     set$KB$clock: proc(Period) ext;
23 2         dcl Period intg;                      /* timeout delay in milliseconds */
24 2         end set$KB$clock;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```

$subtitle('KB: Keyboard Stateware')
/*
 *   KB interrupt entry (level 6)
 */
25 1   kb$irq: proc pub rent;
26 2       do case kb$state;
27 3           /*
28 4               * state 0 to state 1: shift register (full) interrupt
29 4               */
30 4       kbst0: do;
31 4           via(4).ACR= via(4).ACR and not SR$enable; /* disable shift register */
32 4               /* prepare for interrupt on negative edge of KB RDY */
33 4           via(4).PCR= via(4).PCR and not CB1$pos_edge;
34 4           via(4).IER= 80h or CB1$intbit;
35 4           disable;                                /* time critical section */
36 4           kb$data = via(4).SR;                  /* get KB data from SR (clears SR IRQ) */
37 4           via(4).IER= SR$intbit;                 /* disable SR interrupt */
38 4           via(4).RB = via(4).RB or kb$ackctl;    /* assert KB ACK control on interrupt */
39 4           enable;                            /* (CB1 IRQ is reset) */
40 4           kb$state = 1;                      /* end of critical section */
41 4       end;                                /* set to state 1 */
42 4
43 4           /*
44 4               * state 1 to state 2: interrupt from negative edge on KB$RDY
45 4               */
46 4       kbst1: do;
47 4           disable;                                /* time critical section */
48 4           if (via(4).RA and kb$databit) <> 0 then /* if data bit is not low then */
49 4               call kb$error;                    /* stop bit error has occurred */
50 4           else do;                          /* prepare for interrupt on positive edge of KB RDY */
51 4               via(4).PCR= via(4).PCR or CB1$pos_edge;
52 4                   /* release KB ACK control on interrupt */
53 4               via(4).RB = via(4).RB and not kb$ackctl; /* (CB1 IRQ is reset) */
54 4               kb$state = 2;                      /* set to state 2 */
55 4           end;                                /* end of critical section */
56 4       enable;                            /* end; */
57 4

```

**\$eject**

```
/*
 * state 2 to state 0: interrupt from positive edge on KB$RDY
 */
49 3    kbst2: do;
50 4        if (via(4).RA and kb$databit) = 0 then      /* if data bit is low then      */
51 4            call kb$error;                      /* stop bit error has occurred  */
52 4        else do;
53 5            call kb$reset;                  /*reset hardware/software for next event */
54 5                /* call event processing routine with order of bits reversed to */
55 5                /* reflect physical key number and event type (open or close) */
56 5                if not Process$Event( shl(Ctable(kb$data and 0fh),4)
57 5                    or Ctable(shr(kb$data,4)) ) then
58 5                    call signal$KB$error;          /* signal error in event process */
59 5                end;
56 5            end;
58 3        end;
59 2    end kb$irq;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```

$subtitle("KB: Keyboard support routines")

60 1 kb$reset: proc rent;                                /* puts KB hardware/software into state 0 */
61 2     dcl dummy byte;

62 2         via(4).IER = CB1$intbit;
63 2         via(4).RB = via(4).RB and not kb$ackctl;
64 2         via(4).ACR = via(4).ACR or SR$enable;
65 2         dummy = via(4).SR;
66 2         via(4).IER = 80h or SR$intbit;
67 2         kb$state = 0;
68 2         call set$KB$clock(0);
69 2     end kb$reset;

70 1 kb$error: proc rent;                               /* clear CB1 interrupts */
71 2     via(4).RB = via(4).RB or kb$ackctl;           /* release kb$ack */
72 2     via(4).IER = 7fh;                            /* enable shft reg */
73 2     call set$KB$clock(kb$TIMEOUT);             /* clr any pending irq */
74 2     end kb$error;                           /* enable sr interrupts */
                                                /* init keybrd state */
                                                /* clear timeout counter */

75 1 kb$init: proc pub rent;                          /* force kb$ack high */
76 2     via(4).RB = via(4).RB and (OFFh-3);          /* allow no interrupts */
77 2     via(4).DDRA = via(4).DDRA and not kb$dataabit;
78 2     via(4).DDRB = via(4).DDRB or kb$ackctl;       /* time out keyboard */
79 2     via(4).IER = 7fh;
80 2     via(4).PCR = 0;
81 2     via(4).ACR = 0;

82 2     via(2).ACR= (via(2).ACR and 0c0h) or 40h;
83 2     via(2).timer1L= tick*1000;
84 2     via(2).IER = timer1_ena and 7fh;
85 2     call kb$reset;
86 2     end kb$init;

```

\$SUBTITLE ("CRTreg: controller chip registers")

```
87 1 DCL CRT$0 byte AT (OE8000H); /* CRT-chip address register */
88 1 DCL CRT$1 BYTE AT (OE8001H); /* CRT-chip internal register port */

/*
 *      Set CRT register
 */
89 1 set$CRT$reg: proc (reg,value) rent;
90 2     dcl reg byte;
91 2     dcl value byte;
92 2         CRT$0= reg; /* select register */
93 2         CRT$1= value; /* set data */
94 2     end set$CRT$reg;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

\$SUBTITLE ('CRTreg: cursor-display mode control')

```
95 1     dcl  rast$start lit   '10';           /* CRT reg: cursor-start & cursor-display mode */

96 1     DCL  Cursor$PAR  BYTE; /* VAR: contents for CRT cursor-start raster & cursor display mode */
97 1     dcl  blink$on    boolean;             /* FLAG: =0 Blinking cursor on (fast) */
98 1     dcl  curs$off    boolean;             /* FLAG: <>0 Cursor off */

/*
 *      Set cursor to current Cursor parameter byte.
 */
99 1     set$cursor: proc rent;
100 2       call set$CRT$reg(rast$start,Cursor$PAR); /* set raster start reg */
101 2       end set$cursor;

/*
 *      Set block cursor.
 */
102 1     BLOCK$CRS:PROC RENT;
103 2       Cursor$PAR = Cursor$PAR AND 0E0h;          /* set block cursor */
104 2       call set$cursor;                          /* set cursor mode reg */
105 2       END BLOCK$CRS;

/*
 *      Set underscore cursor.
 */
106 1     UNDERSCORE$CRS:PROC RENT;
107 2       Cursor$PAR = 00Fh OR (Cursor$PAR AND 0E0h); /* set underscore cursor */
108 2       call set$cursor;                          /* set cursor mode reg */
109 2       END UNDERSCORE$CRS;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```

$eject
/*
 *      Return cursor to previous modes: block or underline, steady or flashing
 */
110 1 CURSOR$ON:PROC RENT;
111 2     curs$off= false;                                /* reset cursor off flag */
112 2     if blink$on then Cursor$par= Cursor$par or 060h; /* set to flashing mode */
114 2     else Cursor$par= Cursor$par and 01Fh;          /* set to steady mode */
115 2     call set$cursor;                               /* set cursor mode reg */
116 2 END CURSOR$ON;
/*
 *      Turn cursor off.
 */
117 1 CURSOR$OFF:PROC RENT;
118 2     curs$off= true;                                /* set cursor off flag */
119 2     Cursor$PAR = 020h OR (Cursor$PAR AND 01Fh);    /* set to off mode */
120 2     call set$cursor;                               /* set cursor mode reg */
121 2 END CURSOR$OFF;

/*
 *      Set cursor blinking.
 */
122 1 CRS$BLINK$ON:PROC RENT;
123 2     blink$on= true;                                /* set blinking on flag */
124 2     if not curs$off then Cursor$PAR= 060h OR Cursor$PAR; /* set flashing,if not off */
126 2     call set$cursor;                               /* set cursor mode reg */
127 2 END CRS$BLINK$ON;
/*
 *      Set cursor steady.
 */
128 1 CRS$BLINK$OFF:PROC RENT;
129 2     blink$on= false;                               /* reset blinking on flag */
130 2     if not curs$off then Cursor$PAR= 01Fh and Cursor$PAR; /* set steady,if not off */
132 2     call set$cursor;                               /* set cursor mode reg */
133 2 END CRS$BLINK$OFF;

```

**\$SUBTITLE ("CRTreg: Cursor positioning")**

```
134 1     dcl  cursaddrH  lit   '14'; /* CRT reg: MSByte of cursor location word, bits: xx54$3210 */
135 1     dcl  cursaddrL  lit   '15'; /* CRT reg: LSByte of cursor location word */

/*
 *  Position Cursor to Absolute Font Cell number
 *      and display bank
 */

136 1     POS$Cursor: proc (Cell$number) pub rent;
137 2         dcl Cell$Number word;           /* Absolute Font Cell Number & display bank */
138 2         call set$CRT$reg (cursaddrL, low(Cell$number));
139 2         call set$CRT$reg (cursaddrH, high(Cell$number));
140 2     end POS$Cursor;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

\$SUBTITLE ("CRT: video contrast & brightness")

```

141 1     DCL   CBctrl    BYTE   AT (0E8040H);      /* Contrast & Brightness control register */
                                                       /* bits: CCCB$BB-- */

142 1     /*
143 2       *      Raise video contrast one level.
144 2       */
145 2     contrast$up: proc rent;
146 2       dcl a byte;
147 2       if (a:= (CBctrl + 20h) and 0E0h) <> 0 then      /* add & check upper limit */
148 2         CBctrl= (CBctrl and 01FH) or a;           /* set contrast, bits: 765 */
149 2     end contrast$up;
150 2     /*
151 2       *      Lower video contrast one level.
152 2       */
153 2     contrast$down: proc rent;
154 2       dcl a byte;
155 2       if (a:= (CBctrl - 20h) and 0E0h) <> 0E0h then    /* sub & check lower limit */
156 2         CBctrl= (CBctrl and 01FH) or a;           /* set contrast, bits: 765 */
157 2     end contrast$down;

158 2     /*
159 2       *      Raise video brightness one level.
160 2       */
161 2     bright$up: proc rent;
162 2       dcl a byte;
163 2       if (a:= (CBctrl + 4) and 01CH) <> 0 then      /* add & check upper limit */
164 2         CBctrl= (CBctrl and 0E3H) or a;           /* set brightness, bits: 432 */
165 2     end bright$up;
166 2     /*
167 2       *      Lower video brightness one level.
168 2       */
169 2     bright$down: proc rent;
170 2       dcl a byte;
171 2       if (a:= (CBctrl - 4) and 01Ch) <> 01Ch then    /* sub & check lower limit */
172 2         CBctrl= (CBctrl and 0E3H) or a;           /* set brightness, bits: 432 */
173 2     end bright$down;

```

## \$SUBTITLE ("CRT: display RAM/Font Cells")

```

162 1 dcl screen$ram word at (0F0000h);      /* memory address of display RAM */
163 1 dcl screen$addr ptr;                  /* display ram pointer, base of word ARRAY */
164 1 DCL SCREEN based screen$addr (2000) word; /* ARRAY of Font Cell Pointers */

/*
 * Screen Buffer Word variables
 */

165 1 dcl char$mode word pub;           /* CRT attribute bits: 7654$3---- */
166 1 dcl char$base word pub;          /* CRT Font Cell Pointer base for */
                                         /* ASCII symbol index */

167 1 DCL REVBIT LIT '8000H';
168 1 DCL BGBIT LIT '4000H';
169 1 DCL UNDBIT LIT '2000H';
170 1 dcl INVBIT lit '1000h';
171 1 dcl extraBIT lit '0800h';

/*
 * Display symbol from character set (typically ASCII)
 * at absolute Font Cell number
 * (typically: <line> * <display width> + <column> )
 * with current Cursor & Display modes.
 */

172 1 Display$symbol: proc (Symbol$code,Cell$number) pub rent;
173 2     dcl Symbol$code byte;             /* Symbol print code */
174 2     dcl Cell$Number word;            /* Absolute Font Cell Number */
175 2     screen(Cell$Number)= (Symbol$code + char$base) OR char$mode;
176 2 end Display$symbol;

```

```
$SUBTITLE ("CRT hardware initialization")  
/* COMMENT THIS !!!! */  
177 1 DCL CRT$config (*) BYTE DATA (92,80, 81,OCFh, 25,6, 25,25, 3,14, 0,15, 0,0, 0,0);  
  
178 1 CRT$Init: PROC;  
179 2     DCL I BYTE;  
  
screen$ram;  
  
181 2     char$mode= BGBIT;  
182 2     char$base= 20;  
  
183 2     curs$off= false;  
184 2     blink$on= false;  
185 2     Cursor$PAR= 0;  
  
186 2     DO I=0 TO OFH;  
187 3     CALL SET$CRT$REG (I,(CRTconfig(I)));  
188 3     END;  
  
189 2     END CRT$Init;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

## \$SUBTITLE ("SOUND variables &amp; hardware defs")

```
190 1 dcl bell$freq LIT    '76';           /* period of bell tone: frequency= 14.9KHz */
191 1 dcl codec$clk     word at (0E8084h);   /* TIMER1: codec clock frequency      */
192 1 dcl codec$ctl     byte at (0E808Bh);   /* ACR: codec clock control register */
193 1 dcl codec$sda     word at (0E8060h);   /* */
194 1 dcl volume        byte at (0E802Ah);   /* SR: volume shift-register          */
195 1 dcl vol$ctl       byte at (0E802Bh);   /* ACR: SR contrl register           */
196 1 dcl vol$clk       word at (0E8028h);   /* TIMER2: volume SR clock           */
197 1 dcl bell$on        byte;                 /* FLAG: bell sound presently active */
198 1 dcl vol$level      byte;                 /*current volume level (nine levels: 0 --> 8)
199 1 dcl vol$table (*) byte data (0FFh,7FH,3FH,1FH,0FH,7,3,1,0);           /* volume shift pattern lookup table */
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

\$SUBTITLE ('SOUND: Bell control')

```
/*
 * Software clock resource -- set timeout for interrupt to Bell$clock
 */

200 1    set$BELL$clock: proc (Period) ext;
201 2        dcl Period intg;                      /* timeout delay in milliseconds */
202 2        end set$BELL$clock;

/*
 * CODEC Hardware reset
 */

203 1    Bell$init: proc pub rent;
204 2        vol$level= length(vol$table)-2;
205 2        call Bell$clock;                     /* set initial volume level near max */
206 2        end Bell$init;                      /* set hardware to a known & quiet state */
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

**\$eject**

```

207 1     Bell$clock: proc pub rent;
208 2         codec$ctl = codec$ctl and not 0C0h;           /* disable codec clock */
209 2         codec$sda = 5E00h;                            /* initialize codec SDA to input mode... */
210 2         codec$sda = 0D40h;                            /* ... to reduce extraneous noise */
211 2         codec$sda = 0AA80h;
212 2         codec$sda = 00C0h;

213 2         vol$ctl = (vol$ctl and not 3Ch) or 10h;      /* set SR & T2 volume register modes */
214 2         vol$clk = 1;                                /* volume clock frequency set beyond perception */
215 2         volume = vol$table(vol$level);            /* set volume to current level */
216 2         bell$on = false;                           /* set bell state to off */

217 2     end bell$clock;

218 1     Ring$bell: proc pub rent;
219 2         if not bell$on then do;
220 3             call bell$clock;
221 3             codec$sda = 0f80h;
222 3
223 3             codec$ctl ^= codec$ctl or 0c0h;
224 3             codec$clk = bell$freq;
225 3             bell$on = true;
226 3             end;
227 2             call set$bell$clock(100);
228 2         end;

```

/\* start bell if sound is off \*/  
 /\* init codec hardware on every bell \*/  
 /\* set output waveform to 4 up & 4 down, \*/  
 /\* a low amplitude triangle wave. \*/  
 /\* set codec clock to free run \*/  
 /\* set audio pitch frequency \*/  
 /\* set bell state on \*/  
 /\* turn off bell in 100 milliseconds \*/

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$SUBTITLE ("SOUND: volume control")
/*
 *      Raise CODEC volume one level.
 */
229 1    volume$up: proc rent;
230 2        if vol$level >= length(vol$table)-1 then      /* check upper limit */
231 2            vol$level= length(vol$table)-1;          /* set to max volume */
232 2            else vol$level= vol$level+1;             /* bump level up by one */
233 2            volume= vol$table(vol$level);           /* set volume register */
234 2        end volume$up;

/*
 *      Lower CODEC volume one level.
 */
235 1    volume$down: proc rent;
236 2        if vol$level >= length(vol$table)-1 then      /* check upper limit */
237 2            vol$level= length(vol$table)-2;          /* set to max volume-1 */
238 2            else
239 2                if vol$level<>0 then vol$level= vol$level-1; /* drop level by one */
240 2                volume= vol$table(vol$level);           /* set volume register */
241 2        end volume$down;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```

$subtitle('SIO: Serial I/O dtrs for TTY: and ULL:')

242   1      /*ctr device dcls*/
        dcl sioctr struc
          (adata byte,
           bdata byte,
           xxx  byte,
           ctrctl byte) at (0E0020h);
        /*sio device dcls*/
243   1      dcl siodev struc
          (adata byte,
           bdata byte,
           actl  byte,
           bctl  byte) at(0E0040h);
244   1      dcl rx$avail literally '1',
              tx$empty literally '4';

245   1      dcl serial_params struc
          (actrlsb byte,
           actrmsb byte,
           bctrlsb byte,
           bctrmsb byte,
           /* if <baud> then lsb = ??h msb = ??h 1.25Mhz/(<baud>*16)
              */                         /*LSByte of chan a.'s baud rate */
           /*MSByte ... */                  /* */
           /*LSByte of chan b.'s baud rate */
           /*MSByte ... */                  /* */
              50 --->      1Ah      06h      50.00 -0- (min.tol.dist.43.75%)
              75 --->      11h      04h      75.00 -0- ("        "    43.75%)
              110 --->     C6h      02h     110.00 -0- ("        "    43.75%)
              134.5 --->    44h      02h     134.00 -0.37% ("        "    40.23%)
              150 --->     08h      02h     150.00 -0- ("        "    43.75%)
              200 --->     86h      01h     200.00 -0- ("        "    43.75%)
              300 --->     04h      01h     300.00 -0- ("        "    43.75%)
              600 --->     82h      00h     600.00 -0- ("        "    43.75%)
              1.2k --->    41h      00h    1201.00 +0.08% ("        "    42.99%)

```

\$eject

|            |     |     |          |         |                       |   |         |
|------------|-----|-----|----------|---------|-----------------------|---|---------|
|            | 2Ch | 00h | 1775.00  | -1.39%  | (                     | " | 30.54%) |
| 1.8k --->  | 2Bh | 00h | 1816.00  | +0.09%  | (                     | " | 42.88%) |
|            | 28h | 00h | 1953.00  | -2.36%  | (                     | " | 21.33)  |
| 2.0k --->  | 27h | 00h | 2003.00  | +0.15%  | (                     | " | 42.32)  |
|            | 21h | 00h | 2367.00  | -1.38%  | (                     | " | 30.64%) |
| 2.4k --->  | 20h | 00h | 2441.00  | +1.71%  | (                     | " | 27.51%) |
|            | 16h | 00h | 3551.00  | -1.36%  | (                     | " | 30.83%) |
| 3.6k --->  | 15h | 00h | 3720.00  | +3.33%  | (                     | " | 12.4%)  |
|            | 11h | 00h | 4595.00  | -4.27%  | (                     | " | 3.185%) |
| 4.8k --->  | 10h | 00h | 4882.00  | +1.02%  | (                     | " | 34.06%) |
|            | 09h | 00h | 8680.55  | -9.58%  | (DISTORTED)           |   |         |
| 9.6k --->  | 08h | 00h | 9765.56  | +1.73%  | (min.tol.dist.27.32%) |   |         |
|            | 06h | 00h | 13020.83 | -9.58%  | (DISTORTED)           |   |         |
|            | 05h | 00h | 15625.00 | +8.51%  | (DISTORTED)           |   |         |
|            | 05h | 00h | 15625.00 | -18.62% | of 19.2k (DISTORTED)  |   |         |
| 19.2k ---> | 04h | 00h | 19531.25 | +1.02%  | (min.tol.dist.34.06%) |   |         |

min.tol.dist. figure assumes no channel noise effects.

NOTE: possible noise DOES NOT includes bias distortion  
caused by various cable capacitance effects\*/

NOTE: Sample Programs only - Not BIOS Reference Listings!

**\$eject**

```
cr2a    byte,      /*bus interface option: 10h if baud a <= baud b
                           14h if baud a > baud b*/
```

```
cr4a    byte,
cr4b    byte,
/*cr4x (16x)$54$(stops)$(even)$(parenb) = 47h
```

```
01 00 ss e p
```

```
ss = 01 1 stop
```

```
= 10 1.5 stop
```

```
= 11 2 stop
```

```
e = 1 even
```

```
e = 0 odd, byte transparent
```

```
p = 1 even or odd
```

```
p = 0 byte transparent*/
```

H

```
cr3a    byte,
```

```
cr3b    byte,
```

```
/*cr3x (rbits)$(autoenb)$4$3$2$1$(renb) = ?1h
```

```
bb 1 0 0 0 0 1
```

```
bb = 11 byte transparent cr3x = E1h
```

```
- 01 even,odd cr3x = 61h*/
```

```
cr5a    byte,
```

```
cr5b    byte) EXT;
```

```
/*cr5x (dtr)$(tbits)$(br)$(tenb)$2$(rts)$0 = ?Ah
```

```
1 bb 0 1 0 1 0
```

```
bb = 11 space,mark cr5x = EAh
```

```
bb = 01 even,odd,no cr5x = AAh*/
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

\$subtitle('SIO: Serial I/O dvr\$ for port A -- TTY\$INSTAT & TTY\$STAT')

```
246 1   TTY$in$stat:proc boolean PUB;  
247 2   if ( (siodev.actl AND rx$avail) <> 0)  
248 2       then return(true);  
249 2       return(false);  
250 2   end TTY$in$stat;  
  
251 1   TTY$stat:proc boolean PUB;  
252 2   if ( (siodev.actl AND tx$empty) = 0)  
253 2       then return(true);  
254 2       return(false);  
255 2   end TTY$stat;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$subtitle(`SIO: Serial I/O dvr for port A -- TTY$GET & TTY$PUT`)
```

```
256 1   TTY$get:proc byte PUB;  
  
      /*user must not activate this procedure if siodev chan. a reg. ptr  
       is not set to 0 (only <> 0 if user has been mucking with hardware*/  
257 2   do while( (siodev.actl AND rx$avail) = 0);      /*wait forever till empty */  
258 3   end;  
259 2   return(siodev.adata);                      /*input form 7201 */  
260 2   end TTY$get;  
  
  
261 1   TTY$put:proc(char) PUB;  
262 2   dcl char byte;  
  
      /*user must not activate this procedure if siodev chan. a reg. ptr  
       is not set to 0 (only <> 0 if user has been mucking with hardware*/  
263 2   do while( (siodev.actl AND tx$empty) = 0);      /*wait forever till empty */  
264 3   end;  
265 2   siodev.adata = char;                      /*output a char */  
266 2   return;  
267 2   end TTY$put;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$subtitle('SIO: Serial I/O dvrs for port B -- UL1$STAT & UL1PUT')
```

```
268 1     UL1$stat:proc boolean PUB;  
269 2     if ( (siodev.bctl AND tx$empty) = 0)  
270 2       then return(true);  
271 2       return(false);  
272 2     end UL1$stat;  
  
273 1     UL1$put:proc(char) PUB;  
274 2     dcl char byte;  
  
      /*user must not activate this procedure if siodev chan. b reg. ptr  
       is not set to 0 (only <> 0 if user has been mucking with hardware*/  
275 2     do while( (siodev.bctl AND tx$empty) = 0);           /*wait forever till empty */  
276 3     end;  
277 2     siodev.bdata = char;                            /*output a char */  
278 2     return;  
279 2     end UL1$put;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
280 1   $subtitle(`SIO: Serial I/O dvrs for ports A & B -- SIO$INIT`)
281 2   SIO$init:proc`PUB;
282 2   siodev.actl = 00$011$000b;                                /*chan. a reset */ */
282 2   siodev.bctl = 00$011$000b;                                /*chan. b reset */ */

283 2   /*load timer now; cant touch 7201 chip for 4 2.5Mhz clocks*/
284 2   sioctr.ctrctl = 36h;                                     /*7$(ctra)$(rl)$(mode)$(bin) */ */
285 2   sioctr.adata = serial_params.actrlsb;
286 2   sioctr.adata = serial_params.actrmsb;
287 2   sioctr.ctrctl = 76h;                                     /*7$(ctrb)$(rl)$(mode)$(bin) */ */
288 2   sioctr.bdata = serial_params.bctrlsb;
288 2   sioctr.bdata = serial_params.bctrmsb;

289 2   /*cr2a bus interface option*/
290 2   siodev.actl = 2;                                         /*-->cr4a */ */
290 2   siodev.actl = serial_params.cr2a;

291 2   /*cr4x*/
292 2   siodev.actl = 4;                                         /*-->cr4a */ */
293 2   siodev.bctl = 4;                                         /*-->cr4b */ */
294 2   siodev.bctl = serial_params.cr4b;

295 2   /*cr3x*/
296 2   siodev.actl = 3;                                         /*-->cr3a */ */
297 2   siodev.actl = serial_params.cr3a;
298 2   siodev.bctl = 3;                                         /*-->cr3b */ */
298 2   siodev.bctl = serial_params.cr3b;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$eject

/*cr5x*/
299 2     siodev.actl = 5;                      /*-->cr5a          */
300 2     siodev.actl = serial_params.cr5a;      */
301 2     siodev.bctl = 5;                      /*-->cr5b          */
302 2     siodev.bctl = serial_params.cr5b;      */

/*cr0x reset ext/st intrs to enable modem control sense--> autoenb chans.
   also --> crlx, set intr params*/
303 2     siodev.actl = 00$010$001b;
304 2     siodev.actl = 0;                         /*no intrs          */
305 2     siodev.bctl = 00$010$001b;
306 2     siodev.bctl = 0;                         /*no intrs          */

307 2     end sio$init;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$subtitle ("PPORT -- centronics interface routines")

/*
 * This module implements the initialization, LISTST, and LIST functions
 * for a Centronics-compatible parallel printer interface, using the
 * 6522 VIA chip.
 *
 * Our entry points are named pp$init, LPT$stat, and LPT$put respectively,
 * it's up to our caller to decode the I/O byte and call the appropriate
 * routines.
 */
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```

$eject
308 1 declare pp$base pointer;           /* baseaddr for a 6522 */
309 1 declare pp based pp$base structure (/* 6522 template */
      rb byte,                         /* out-in reg 'b' */
      ra byte,                         /* out-in reg 'a' */
      ddrb byte,                        /* data-direction, reg 'b' */
      ddra byte,                        /* data-direction, reg 'a' */
      tlcl byte,                        /* tl ctr(r)/lat(w) lo */
      tlch byte,                        /* tl ctr hi */
      tlll byte,                        /* tl latch lo */
      tllh byte,                        /* tl latch hi */
      t2cl byte,                        /* t2 ctr(r)/lat(w) lo */
      t2ch byte,                        /* t2 ctr hi */
      sr byte,                          /* shift register */
      acr byte,                         /* auxiliary ctrl reg */
      pcr byte,                         /* peripheral ctrl reg */
      ifr byte,                          /* interrupt flg register */
      ier byte,                          /* interrupt enbl register */
      rax byte,                          /* out-in reg 'a' NO HANDSHAKE */
);
/*
 * Bit definitions for Centronics-style parallel interface, 'vial'.
 */
310 1 declare vial$base literally '0e8020h'; /* baseaddr for this chip */
311 1 declare ds$1 literally '01h';        /* data strobe (pb0) */
312 1 declare pi$h literally '02h';       /* this datum for vfu (pb1) */
313 1 declare bz$h literally '20h';       /* printer busy (pb5) */
314 1 declare ak$1 literally '40h';       /* printer ack (pb6) */
315 1 declare sl$h literally '80h';       /* on-line and no error (pb7) */
/*
 * Bit definitions for multi-use pio, 'via2'.
 */
316 1 declare via2$base literally '0e8040h'; /* baseaddr for this chip */
317 1 declare te$h literally '01h';        /* talk-enable line */
*/

```

NOTE: Sample Programs only - Not BIOS Reference Listings!

```
$eject
/*
 * initial setup for parallel printer port
 * Note we use via2 during this setup to get talk-enable turned on, and
 * thus someone MUST ALREADY HAVE VIA2 INITIALIZED.
 */

318 1 pp$init: procedure public;
319 2     pp$base = via2$base;
320 2     Pp.rb = pp.rb or te$h;
321 2     pp$base = vial$base;
322 2     pp.ra = 0;
323 2     pp.ddra = Offh;
324 2     pp.rb = ds$1;
325 2     pp.ddrb = ds$1 or pi$h;

326 2 end pp$init;
                                         /* point to secondary chip for te */
                                         /* set 'talk enbl' */          */
                                         /* point struc at primary chip */
                                         /* ra is dataport, init with 0's */
                                         /* set all ra bits as outgoing */
                                         /* rb is ctrlport, init no ds/pi */
                                         /* these 2 only are outgoing */
                                         /* cal/ca2 cbl/cb2 not used */
                                         /* timers/shiftreg not used */
```

NOTE: Sample Programs only - Not BIOS REFERENCE LISTINGS!

```
$eject

/*
 * Test status of printer, return true if on-line and not busy, else
 * false. For some reason, the Altos code explicitly deasserted data
 * strobe before testing; we'll assume that this represents an Altos
 * fubar and is not required here.
 */
327 1 LPT$stat: procedure byte public;
328 2     if (pp.rb and (sl$h or bz$h)) = sl$h then return Offh;
330 2     return 0;
331 2 end LPT$stat;

/*
 * Put one character to the printer interface.
 */
332 1 LPT$put: procedure(ch) public;
333 2     declare ch byte;
334 2     do while LPT$stat = 0; end;           /* wait for printer ready */
336 2     pp.ra = ch;                      /* put outgoing char on the port */
337 2     disable;
338 2     pp.rb = pp.rb and not ds$l;       /* assert data strobe */
339 2     pp.rb = pp.rb or ds$l;          /* deassert data strobe */
340 2     enable;
341 2     return;
342 2 end LPT$put;
```

NOTE: Sample Programs only - Not BIOS Reference Listings!

\$SUBTITLE ("Example software drivers for V9000 Hardware")

343 1 end Hardware;

MODULE INFORMATION:

CODE AREA SIZE = 073EH 1854D  
CONSTANT AREA SIZE = 0000H 0D  
VARIABLE AREA SIZE = 0014H 20D  
MAXIMUM STACK SIZE = 000EH 14D  
807 LINES READ  
0 PROGRAM WARNINGS  
0 PROGRAM ERRORS

END OF PL/M-86 COMPIRATION

c

NOTE: Sample Programs only - Not BIOS Reference Listings!

